CCDL 2020

Objective for this notebook analysis:

We’ll use the same gene expression dataset we used in the previous notebook. It is a pre-processed astrocytoma microarray dataset that we performed a set of differential expression analyses on.

More tidyverse resources:
- R for Data Science
- tidyverse documentation
- Cheatsheet of tidyverse data transformation
- Online tidyverse book chapter

Set Up

The tidyverse is a collection of packages that are handy for general data wrangling, analysis, and visualization. Other packages that are specifically handy for different biological analyses are found on Bioconductor. If we want to use a package’s functions we first need to install them.

TODO: Language around Docker container may change depending on RStudio cloud stuff

In our Docker container, we already have the tidyverse and other packages we will use in this workshop installed for you. But if you needed to install it or other packages available on CRAN, you do it using the install.packages() function like this: install.packages("tidyverse").

library(tidyverse)
Registered S3 method overwritten by 'dplyr':
  method           from
  print.rowwise_df     
── Attaching packages ───────────────────────────────────────────────── tidyverse 1.2.1 ──
✔ ggplot2 3.2.0     ✔ purrr   0.3.2
✔ tibble  2.1.3     ✔ dplyr   0.8.3
✔ tidyr   0.8.3     ✔ stringr 1.4.0
✔ readr   1.3.1     ✔ forcats 0.4.0
── Conflicts ──────────────────────────────────────────────────── tidyverse_conflicts() ──
✖ dplyr::filter() masks stats::filter()
✖ dplyr::lag()    masks stats::lag()

Referencing a library’s function with ::

Note that if we had not imported the tidyverse set of packages using library() like above, and we wanted to use a tidyverse function like read_tsv(), we would need to tell R what package to find this function in. To do this, we would use :: to tell R to load in this function from the readr package by using readr::read_tsv(). You will see this :: method of referencing libraries within packages throughout the course. We like to use it in part to remove any ambiguity in which version of a function we are using; it is not too uncommon for different packages to use the same name for very different functions!

Managing directories

Before we can import the data we need, we should double check where R is looking for files, aka the current working directory. We can do this by using the getwd() function, which will tell us what folder we are in.

# Let's check what directory we are in:
getwd()
[1] "/home/rstudio/kitematic/training-modules/intro-to-R-tidyverse"

For Rmd files, the working directory is wherever the file is located, but commands executed in the console may have a different working directory.

We will want to make a directory for our output and we will call this directory: results. But before we create the directory, we should check if it already exists. We will show two ways that we can do this.

First, we can use the dir() function to have R list the files in our working directory.

# Let's check what files are here
dir()
 [1] "00a-rstudio_guide.md"            "00b-debugging_resources.md"     
 [3] "01-introduction-base-R.nb.html"  "01-introduction-base-R.Rmd"     
 [5] "02-intro_to_ggplot2.nb.html"     "02-intro_to_ggplot2.Rmd"        
 [7] "03-intro_to_tidyverse.nb.html"   "03-intro_to_tidyverse.Rmd"      
 [9] "04a-intro_to_R_exercise.nb.html" "data"                           
[11] "diagrams"                        "metadata.tsv"                   
[13] "plots"                           "README.md"                      
[15] "refine.bio"                      "screenshots"                    
[17] "scripts"                        

This shows us there is no folder called “results” yet.

If we want to more pointedly look for “results” in our working directory we can use the dir.exists() function.

# Check if the results directory exists
dir.exists("results")
[1] FALSE

If the above says FALSE that means we will need to create a results directory using the function dir.create().

# Make a directory within the working directory called 'results'
dir.create("results")

After creating the results directory above, let’s re-run dir.exists() to see if now it exists.

# Re-check if the results directory exists
dir.exists("results")
[1] TRUE

We can use the output of dir.exists() to automatically create or hold off on creating a directory by putting this together in an if statement like below. An if statement has two main parts: First, the test, which is an expression that will result in either TRUE or FALSE. This is put in parenthesis immediately after the if. The next part is the body, which is the commands that will be executed if the test is TRUE. These are placed within a set of braces { }. Note that we used an exclamation point in the test to signify that we want a directory to be created only if dir.exists(results) is NOT equal to TRUE.

# If 'results' directory doesn't exist...
if (!dir.exists("results")) {
  # ... create a 'results' directory
  dir.create("results")
}

The dir.exists() function will not work on files themselves. In that case, there is an analogous function called file.exists().

Try using the file.exists() function to see if the file gene_results_GSE44971.tsv exists in the current directory. Use the code chunk we set up for you below. Note that in our notebooks (and sometimes elsewhere), wherever you see a <FILL_IN_THE_BLANK> like in the chunk below, that is meant for you to replace (including the angle brackets) with the correct phrase before you run the chunk (otherwise you will get an error).

# Replace the <PUT_FILE_NAME_HERE> with the name of the file you are looking for
# Remember to use quotes to make it a character string
file.exists(<PUT_FILE_NAME_HERE>)

Now that we’ve determined that gene_results_GSE44971.tsv exists, we are ready to read it into our R environment.

Read a TSV file

Declare the name of the directory where we will read in the data.

data_dir <- "data"

Although base R has functions to read in data files, the functions in the readr package (part of the tidyverse) are faster and more straightforward to use so we are going to use those here. Because the file we are reading in is a TSV (tab separated values) file we will be using the read_tsv function. There are analogous functions for CSV (comma separated values) files (read_csv()) and other files types.

Read in the differential expression analysis results file

stats_df <- readr::read_tsv(file.path(data_dir,
                                      "gene_results_GSE44971.tsv"))
Parsed with column specification:
cols(
  ensembl_id = col_character(),
  gene_symbol = col_character(),
  contrast = col_character(),
  log_fold_change = col_double(),
  avg_expression = col_double(),
  t_statistic = col_double(),
  p_value = col_double(),
  adj_p_value = col_double()
)

Following the template of the previous chunk, use this chunk to read in the file GSE44971.tsv that is in the data folder.

# Use this chunk to read in data from the file `GSE44971.tsv`

Use this chunk to explore what gene_df looks like.

# Explore `gene_df`

What information is contained in gene_df?

dplyr pipes

One nifty feature of the tidyverse is pipes: %>% These handy things allows you to funnel the result of one expression to the next, making your code a little more streamlined.

For example, the output from this:

filter(stats_df, contrast == "male_female")

…is the same as the output from this:

stats_df %>% filter(contrast == "male_female")

This can make your code cleaner and easier to follow a series of related commands. Let’s look at an example with our stats of of how the same functions look with or without pipes:

Example 1: without pipes:

stats_arranged <- arrange(stats_df, t_statistic)
stats_filtered <- filter(stats_arranged, avg_expression > 50)
stats_nopipe <- select(stats_filtered, contrast, log_fold_change, p_value)

UGH, we have to keep track of all of those different intermediate data frames and type their names so many times here! We could maybe streamline things by using the same variable name at each stage, but even then there is a lot of extra typing, and it is easy to get confused about what has been done where. It’s annoying and makes it harder for people to read.

Example 2: Same result as 1 but with pipes!

# Example of the same modifications as above but with pipes!
stats_pipe  <- stats_df %>%
               arrange(t_statistic) %>%
               filter(avg_expression > 50) %>%
               select(contrast, log_fold_change, p_value)

What the %>% (pipe) is doing here is feeding the result of the expression on its left into the first argument of the next function (to its right, or on the next line here). We can then skip that first argument (the data in these cases), and move right on to the part we care about at that step: what we are arranging, filtering, or selecting in this case.

Let’s double check that these are the same by using the function, all.equal().

all.equal(stats_nopipe, stats_pipe)
[1] TRUE

all.equal() is letting us know that these two objects are the same.

Now that hopefully you are convinced that the tidyverse can help you make your code neater and easier to use and read, let’s go through some of the popular tidyverse functions and so we can create pipelines like this.

Common tidyverse functions

Let’s say we wanted to filter this gene expression dataset to particular sample groups. In order to do this, we would use the function filter() as well as a logic statement (usually one that refers to a column or columns in the data frame).

# Here let's filter the data to a particular gene
stats_df %>% 
  filter(gene_symbol == "SNCA")

We can use filter() similarly for numeric statements.

# Here let's filter the data to have average expression values above 50
stats_df %>% 
  filter(avg_expression > 50)

We can apply multiple filters at once, which will require all of them to be satisfied for every row in the results:

stats_df %>% 
  filter(contrast == "male_female", 
         avg_expression > 50)

When we are filtering, the %in% operator can come in handy if we have multiple items we would like to match. Let’s take a look at what using %in% does.

stats_df$gene_symbol %in% c("SNCA", "CDKN1A")
   [1] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [15] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [29] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [43] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [57] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [71] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [85] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
  [99] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [113] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [127] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [141] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [155] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [169] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [183] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [197] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [211] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [225] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [239] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [253] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [267] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [281] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [295] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [309] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [323] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [337] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [351] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [365] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [379] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [393] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [407] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [421] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [435] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [449] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [463] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [477] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [491] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [505] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [519] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [533] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [547] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [561] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [575] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [589] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [603] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [617] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [631] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [645] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [659] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [673] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [687] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [701] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [715] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [729] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [743] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [757] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [771] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [785] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [799] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [813] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [827] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [841] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [855] FALSE FALSE FALSE FALSE  TRUE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [869] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [883] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [897] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [911] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [925] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [939] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [953] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [967] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [981] FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE FALSE
 [995] FALSE FALSE FALSE FALSE FALSE FALSE
 [ reached getOption("max.print") -- omitted 5804 entries ]

%in% returns a logical vector that now we can use in dplyr::filter.

stats_df %>% 
  filter(gene_symbol %in% c("SNCA", "CDKN1A"))

Let’s return to our first filter() and build on to it. This time, let’s keep only some of the columns from the data frame using the select() function. Let’s also save this as a new data frame called stats_filtered_df.

stats_filtered_df <- stats_df %>% 
  filter(contrast == "male_female", 
         avg_expression > 50) %>%
  select(log_fold_change, t_statistic)

Let’s say we wanted to arrange this dataset so that the genes are arranged by the smallest p values to the largest. In order to do this, we would use the function arrange() as well as the column we would like to sort by (in this case p_value).

stats_df %>% 
  arrange(p_value) 

What if we want to sort from largest to smallest? Like if we want to see the genes with the highest average expression? We can use the same function, but instead use the desc() function and now we are using avg_expression column.

stats_df %>% 
  arrange(desc(avg_expression))

What if we would like to create a new column of values? For that we use mutate() function.

stats_df %>% 
  mutate(log10_p_value = -log10(p_value))

What if we want to obtain summary statistics for a column? The summarize function allows us to calculate summary statistics for a column. Here we will use summarize to obtain an mean log folder change over all the genes.

stats_df %>% 
  summarize(mean(log_fold_change))

What if we’d like to obtain a summary statistics but have them for various groups? Conveniently named, there’s a function called group_by() that seamlessly allows us to do this. Also note that group_by() allows us to group by multiple variables at a time if you want to.

stats_summary_df <- stats_df %>%
      group_by(contrast) %>% 
      summarize(mean(log_fold_change))

Let’s look at a preview of what we made:

stats_summary_df

Here we have the mean log fold change expression per each contrast we made.

A brief intro to the apply family of functions

In base R, the apply family of functions can be an alternative methods for performing transformations across a data frame, matrix or other object structures.

One of this family is (shockingly) the function apply(), which operates on matrices or data frames. The first argument to apply() is the data object we want to work on. The third argument is the function we will apply to each row or column of the data object. The second argument in specifies whether we are applying the function across rows or across columns (1 for rows, 2 for columns).

Remember that gene_df is a gene x sample gene expression data frame.

# Calculate row means
gene_means <- apply(gene_df[, -1], 1, mean) # Notice we are using 1 here
Error in apply(gene_df[, -1], 1, mean) : object 'gene_df' not found

Why do we have a [, -1] after gene_df in the above chunk?

Now let’s investigate the same set up, but use 2 to apply over the columns of our matrix.

rr # Calculate sample means sample_means <- apply(gene_df[, -1], 2, mean) # Notice we use 2 here

How long will sample_means be?

length(sample_means)

[1] 58

Although the apply functions may not be as easy to use as the tidyverse functions, for some applications, apply methods can be better suited. In this workshop, we will not delve too deeply into the various other apply functions (tapply(), lapply(), etc.) but you can read more examples about them here.

The dplyr::join functions

Let’s say we have a scenario where we have two data frames that we would like to combine. Recall that stats_df and gene_df are data frames that contain information about some of the same genes. The dplyr::join family of functions are useful for various scenarios of combining data frames.

For now, we will focus on inner_join(), which will combine data frames by only keeping information about matching rows that are in both data frames. We need to use the by argument to designate what column(s) should be used as a key to match the data frames. In this case we want to match the gene information between the two, so we will specify that we want to compare values in the ensembl_id column from stats_df to the Gene column from gene_df.

rr stats_df %>% inner_join(gene_df, by = c(‘ensembl_id’ = ‘Gene’))

Save data to files

Save to TSV files

Let’s write some of the data frames we created to a file. To do this, we can use the readr library of _write() functions. The first argument of write_tsv() is the data we want to write, and the second argument is a character string that describes the path to the new file we would like to create. Remember that we created a results directory to put our output in, but if we want to save our data to a directory other than our working directory, we need to specify this. This is what we will use the file.path() function for. Let’s look in a bit more detail what file.path() does, by examining the results of the function in the examples below.

rr # Which of these file paths is what we want to use to save our data to the # results directory we created at the beginning of this notebook? file.path(-install, _summary.tsv)

[1] \docker-install/stats_summary.tsv\

rr file.path(, _summary.tsv)

[1] \results/stats_summary.tsv\

rr file.path(_summary.tsv, )

[1] \stats_summary.tsv/results\

Replace <NEW_FILE_PATH> below with the file.path() statement from above that will successfully save our file to the results folder

# Write our data frame to a TSV file
readr::write_tsv(stats_summary_df, <NEW_FILE_PATH>)

Check in your results directory to see if your new file has successfully saved.

Save to RDS files

For this example we have been working with data frames, which are conveniently represented as TSV or CSV tables. However, in other situations we may want to save more complicated or very large data structures, RDS (R Data Serialized/Single) files may be a better option for saving our data. RDS is R’s special file format for holding data exactly as you have it in your R environment. RDS files can also be compressed, meaning they will take up less space on your computer. Let’s save our data to an RDS file in our results folder. You will need to replace the .tsv with .RDS, but you can use what we determined as our file path for the last chunk as your template.

# Write your object to an RDS file
readr::write_rds(stats_summary, <PUT_CORRECT_FILE_PATH_HERE>)

Read an RDS file

Since now you have learned the readr functions: read_tsv(), write_tsv(), and now, write_rds(), what do you suppose the function you will need to read your RDS file is called? Use that function here to re-import your data in the chunk we set up for you below.

# Read in your RDS file
reimport_df <- <PUT_FUNCTION_NAME>(file.path("results", "stats_clean.RDS"))

As is good practice, we will end this session by printing out our session info.

Session Info

rr # Print out the versions and packages we are using in this session sessionInfo()

R version 3.6.0 (2019-04-26)
Platform: x86_64-pc-linux-gnu (64-bit)
Running under: Debian GNU/Linux 9 (stretch)

Matrix products: default
BLAS/LAPACK: /usr/lib/libopenblasp-r0.2.19.so

locale:
 [1] LC_CTYPE=en_US.UTF-8       LC_NUMERIC=C              
 [3] LC_TIME=en_US.UTF-8        LC_COLLATE=en_US.UTF-8    
 [5] LC_MONETARY=en_US.UTF-8    LC_MESSAGES=C             
 [7] LC_PAPER=en_US.UTF-8       LC_NAME=C                 
 [9] LC_ADDRESS=C               LC_TELEPHONE=C            
[11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C       

attached base packages:
[1] stats     graphics  grDevices utils     datasets  methods   base     

other attached packages:
 [1] patchwork_1.0.0.9000 maftools_2.2.10      forcats_0.4.0       
 [4] stringr_1.4.0        dplyr_0.8.3          purrr_0.3.2         
 [7] readr_1.3.1          tidyr_0.8.3          tibble_2.1.3        
[10] ggplot2_3.2.0        tidyverse_1.2.1     

loaded via a namespace (and not attached):
 [1] tidyselect_0.2.5   wordcloud_2.6      xfun_0.8          
 [4] splines_3.6.0      haven_2.1.1        lattice_0.20-38   
 [7] colorspace_1.4-1   generics_0.0.2     htmltools_0.3.6   
[10] yaml_2.2.0         base64enc_0.1-3    survival_2.44-1.1 
[13] rlang_0.4.0        pillar_1.4.2       glue_1.3.1        
[16] withr_2.1.2        RColorBrewer_1.1-2 modelr_0.1.4      
[19] readxl_1.3.1       munsell_0.5.0      gtable_0.3.0      
[22] cellranger_1.1.0   rvest_0.3.4        evaluate_0.14     
[25] knitr_1.23         broom_0.5.2        Rcpp_1.0.1        
[28] backports_1.1.4    scales_1.0.0       jsonlite_1.6      
[31] hms_0.4.2          digest_0.6.20      stringi_1.4.3     
[34] grid_3.6.0         cli_1.1.0          tools_3.6.0       
[37] magrittr_1.5       lazyeval_0.2.2     crayon_1.3.4      
[40] pkgconfig_2.0.2    Matrix_1.2-17      data.table_1.12.2 
[43] xml2_1.2.0         lubridate_1.7.4    assertthat_0.2.1  
[46] rmarkdown_1.13     httr_1.4.0         rstudioapi_0.10   
[49] R6_2.4.0           nlme_3.1-140       compiler_3.6.0    
LS0tCnRpdGxlOiAiSW50cm8gdG8gdGlkeXZlcnNlIgpvdXRwdXQ6ICAgCiAgaHRtbF9ub3RlYm9vazogCiAgICB0b2M6IHRydWUKICAgIHRvY19mbG9hdDogdHJ1ZQotLS0KCioqQ0NETCAyMDIwKioKCiMjIE9iamVjdGl2ZSBmb3IgdGhpcyBub3RlYm9vayBhbmFseXNpczogCgpXZSdsbCB1c2UgdGhlIHNhbWUgZ2VuZSBleHByZXNzaW9uIGRhdGFzZXQgd2UgdXNlZCBpbiB0aGUgW3ByZXZpb3VzIG5vdGVib29rXSguLzAyLWludHJvX3RvX2dncGxvdDIuUm1kKS4KSXQgaXMgYSBwcmUtcHJvY2Vzc2VkIFthc3Ryb2N5dG9tYSBtaWNyb2FycmF5IGRhdGFzZXRdKGh0dHBzOi8vd3d3LnJlZmluZS5iaW8vZXhwZXJpbWVudHMvR1NFNDQ5NzEvZ2VuZS1leHByZXNzaW9uLWRhdGEtZnJvbS1waWxvY3l0aWMtYXN0cm9jeXRvbWEtdHVtb3VyLXNhbXBsZXMtYW5kLW5vcm1hbC1jZXJlYmVsbHVtLWNvbnRyb2xzKSAKdGhhdCB3ZSBwZXJmb3JtZWQgYSBzZXQgb2YgW2RpZmZlcmVudGlhbCBleHByZXNzaW9uIGFuYWx5c2VzIG9uXSguL3NjcmlwdHMvMDAtc2V0dXAtaW50cm8tdG8tUi5SKS4gIAoKKipNb3JlIHRpZHl2ZXJzZSByZXNvdXJjZXM6KiogIAotIFtSIGZvciBEYXRhIFNjaWVuY2VdKGh0dHBzOi8vcjRkcy5oYWQuY28ubnovKSAgCi0gW3RpZHl2ZXJzZSBkb2N1bWVudGF0aW9uXShodHRwczovL2RwbHlyLnRpZHl2ZXJzZS5vcmcvKSAgCi0gW0NoZWF0c2hlZXQgb2YgdGlkeXZlcnNlIGRhdGEgdHJhbnNmb3JtYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9yc3R1ZGlvL2NoZWF0c2hlZXRzL3Jhdy9tYXN0ZXIvZGF0YS10cmFuc2Zvcm1hdGlvbi5wZGYpICAKLSBbT25saW5lIHRpZHl2ZXJzZSBib29rIGNoYXB0ZXJdKGh0dHBzOi8vcHJpdmVmbC5naXRodWIuaW8vYWR2cjM4Ym9vay90aWR5dmVyc2UuaHRtbCkgIAoKIyMgU2V0IFVwCgpUaGUgdGlkeXZlcnNlIGlzIGEgY29sbGVjdGlvbiBvZiBwYWNrYWdlcyB0aGF0IGFyZSBoYW5keSBmb3IgZ2VuZXJhbCBkYXRhIAp3cmFuZ2xpbmcsIGFuYWx5c2lzLCBhbmQgdmlzdWFsaXphdGlvbi4gCk90aGVyIHBhY2thZ2VzIHRoYXQgYXJlIHNwZWNpZmljYWxseSBoYW5keSBmb3IgZGlmZmVyZW50IGJpb2xvZ2ljYWwgYW5hbHlzZXMgYXJlIApmb3VuZCBvbiBbQmlvY29uZHVjdG9yXShodHRwczovL3d3dy5iaW9jb25kdWN0b3Iub3JnLykuCklmIHdlIHdhbnQgdG8gdXNlIGEgcGFja2FnZSdzIGZ1bmN0aW9ucyB3ZSBmaXJzdCBuZWVkIHRvIGluc3RhbGwgdGhlbS4KClRPRE86IExhbmd1YWdlIGFyb3VuZCBEb2NrZXIgY29udGFpbmVyIG1heSBjaGFuZ2UgZGVwZW5kaW5nIG9uIFJTdHVkaW8gY2xvdWQgc3R1ZmYKCkluIG91ciBEb2NrZXIgY29udGFpbmVyLCB3ZSBhbHJlYWR5IGhhdmUgdGhlIGB0aWR5dmVyc2VgIGFuZCBvdGhlciBwYWNrYWdlcyB3ZSB3aWxsCnVzZSBpbiB0aGlzIHdvcmtzaG9wIGluc3RhbGxlZCBmb3IgeW91LiAKQnV0IGlmIHlvdSBuZWVkZWQgdG8gaW5zdGFsbCBpdCBvciBvdGhlciBwYWNrYWdlcyBhdmFpbGFibGUgb24gQ1JBTiwgeW91IApkbyBpdCB1c2luZyB0aGUgYGluc3RhbGwucGFja2FnZXMoKWAgZnVuY3Rpb24gbGlrZSB0aGlzOiAKYGluc3RhbGwucGFja2FnZXMoInRpZHl2ZXJzZSIpYC4KCmBgYHtyIExvYWQgdGhlIHRpZHl2ZXJzZX0KbGlicmFyeSh0aWR5dmVyc2UpCmBgYAoKIyMjIFJlZmVyZW5jaW5nIGEgbGlicmFyeSdzIGZ1bmN0aW9uIHdpdGggYDo6YAoKTm90ZSB0aGF0IGlmIHdlIGhhZCBub3QgaW1wb3J0ZWQgdGhlIHRpZHl2ZXJzZSBzZXQgb2YgcGFja2FnZXMgdXNpbmcgYGxpYnJhcnkoKWAgCmxpa2UgYWJvdmUsIGFuZCB3ZSB3YW50ZWQgdG8gdXNlIGEgdGlkeXZlcnNlIGZ1bmN0aW9uIGxpa2UgYHJlYWRfdHN2KClgLCB3ZSAKd291bGQgbmVlZCB0byB0ZWxsIFIgd2hhdCBwYWNrYWdlIHRvIGZpbmQgdGhpcyBmdW5jdGlvbiBpbi4KVG8gZG8gdGhpcywgd2Ugd291bGQgdXNlIGA6OmAgdG8gdGVsbCBSIHRvIGxvYWQgaW4gdGhpcyBmdW5jdGlvbiBmcm9tIHRoZSAKYHJlYWRyYCBwYWNrYWdlIGJ5IHVzaW5nIGByZWFkcjo6cmVhZF90c3YoKWAuCllvdSB3aWxsIHNlZSB0aGlzIGA6OmAgbWV0aG9kIG9mIHJlZmVyZW5jaW5nIGxpYnJhcmllcyB3aXRoaW4gcGFja2FnZXMgCnRocm91Z2hvdXQgdGhlIGNvdXJzZS4gCldlIGxpa2UgdG8gdXNlIGl0IGluIHBhcnQgdG8gcmVtb3ZlIGFueSBhbWJpZ3VpdHkgaW4gd2hpY2ggdmVyc2lvbiBvZiBhIApmdW5jdGlvbiB3ZSBhcmUgdXNpbmc7IGl0IGlzIG5vdCB0b28gdW5jb21tb24gZm9yIGRpZmZlcmVudCBwYWNrYWdlcyB0byB1c2UgdGhlIApzYW1lIG5hbWUgZm9yIHZlcnkgZGlmZmVyZW50IGZ1bmN0aW9ucyEKCiMjIE1hbmFnaW5nIGRpcmVjdG9yaWVzCgpCZWZvcmUgd2UgY2FuIGltcG9ydCB0aGUgZGF0YSB3ZSBuZWVkLCB3ZSBzaG91bGQgZG91YmxlIGNoZWNrIHdoZXJlIFIgaXMgCmxvb2tpbmcgZm9yIGZpbGVzLCBha2EgdGhlIGN1cnJlbnQgKip3b3JraW5nIGRpcmVjdG9yeSoqLiAKV2UgY2FuIGRvIHRoaXMgYnkgdXNpbmcgdGhlIGBnZXR3ZCgpYCBmdW5jdGlvbiwgd2hpY2ggd2lsbCB0ZWxsIHVzIHdoYXQgZm9sZGVyCndlIGFyZSBpbi4gCgpgYGB7cn0KIyBMZXQncyBjaGVjayB3aGF0IGRpcmVjdG9yeSB3ZSBhcmUgaW46CmdldHdkKCkKYGBgCgpGb3IgUm1kIGZpbGVzLCB0aGUgd29ya2luZyBkaXJlY3RvcnkgaXMgd2hlcmV2ZXIgdGhlIGZpbGUgaXMgbG9jYXRlZCwgYnV0IApjb21tYW5kcyBleGVjdXRlZCBpbiB0aGUgY29uc29sZSBtYXkgaGF2ZSBhIGRpZmZlcmVudCB3b3JraW5nIGRpcmVjdG9yeS4KCldlIHdpbGwgd2FudCB0byBtYWtlIGEgZGlyZWN0b3J5IGZvciBvdXIgb3V0cHV0IGFuZCB3ZSB3aWxsIGNhbGwgdGhpcyBkaXJlY3Rvcnk6IApgcmVzdWx0c2AuIApCdXQgYmVmb3JlIHdlIGNyZWF0ZSB0aGUgZGlyZWN0b3J5LCB3ZSBzaG91bGQgY2hlY2sgaWYgaXQgYWxyZWFkeSBleGlzdHMuIApXZSB3aWxsIHNob3cgdHdvIHdheXMgdGhhdCB3ZSBjYW4gZG8gdGhpcy4gCgpGaXJzdCwgd2UgY2FuIHVzZSB0aGUgYGRpcigpYCBmdW5jdGlvbiB0byBoYXZlIFIgbGlzdCB0aGUgZmlsZXMgaW4gb3VyIHdvcmtpbmcgCmRpcmVjdG9yeS4gCgpgYGB7cn0KIyBMZXQncyBjaGVjayB3aGF0IGZpbGVzIGFyZSBoZXJlCmRpcigpCmBgYAoKVGhpcyBzaG93cyB1cyB0aGVyZSBpcyBubyBmb2xkZXIgY2FsbGVkICJyZXN1bHRzIiB5ZXQuIAoKSWYgd2Ugd2FudCB0byBtb3JlIHBvaW50ZWRseSBsb29rIGZvciAicmVzdWx0cyIgaW4gb3VyIHdvcmtpbmcgZGlyZWN0b3J5IHdlIGNhbiAKdXNlIHRoZSBgZGlyLmV4aXN0cygpYCBmdW5jdGlvbi4KCmBgYHtyfQojIENoZWNrIGlmIHRoZSByZXN1bHRzIGRpcmVjdG9yeSBleGlzdHMKZGlyLmV4aXN0cygicmVzdWx0cyIpCmBgYAoKSWYgdGhlIGFib3ZlIHNheXMgYEZBTFNFYCB0aGF0IG1lYW5zIHdlIHdpbGwgbmVlZCB0byBjcmVhdGUgYSBgcmVzdWx0c2AgCmRpcmVjdG9yeSB1c2luZyB0aGUgZnVuY3Rpb24gYGRpci5jcmVhdGUoKWAuCgpgYGB7cn0KIyBNYWtlIGEgZGlyZWN0b3J5IHdpdGhpbiB0aGUgd29ya2luZyBkaXJlY3RvcnkgY2FsbGVkICdyZXN1bHRzJwpkaXIuY3JlYXRlKCJyZXN1bHRzIikKYGBgCgpBZnRlciBjcmVhdGluZyB0aGUgcmVzdWx0cyBkaXJlY3RvcnkgYWJvdmUsIGxldCdzIHJlLXJ1biBgZGlyLmV4aXN0cygpYCB0byBzZWUgCmlmIG5vdyBpdCBleGlzdHMuCgpgYGB7cn0KIyBSZS1jaGVjayBpZiB0aGUgcmVzdWx0cyBkaXJlY3RvcnkgZXhpc3RzCmRpci5leGlzdHMoInJlc3VsdHMiKQpgYGAKCldlIGNhbiB1c2UgdGhlIG91dHB1dCBvZiBgZGlyLmV4aXN0cygpYCB0byBhdXRvbWF0aWNhbGx5IGNyZWF0ZSBvciBob2xkIG9mZiBvbiAKY3JlYXRpbmcgYSBkaXJlY3RvcnkgYnkgcHV0dGluZyB0aGlzIHRvZ2V0aGVyIGluIGFuIGBpZmAgc3RhdGVtZW50IGxpa2UgYmVsb3cuIApBbiBgaWZgIHN0YXRlbWVudCBoYXMgdHdvIG1haW4gcGFydHM6CkZpcnN0LCB0aGUgdGVzdCwgd2hpY2ggaXMgYW4gZXhwcmVzc2lvbiB0aGF0IHdpbGwgcmVzdWx0IGluIGVpdGhlciBgVFJVRWAgb3IgYEZBTFNFYC4KVGhpcyBpcyBwdXQgaW4gcGFyZW50aGVzaXMgaW1tZWRpYXRlbHkgYWZ0ZXIgdGhlIGBpZmAuClRoZSBuZXh0IHBhcnQgaXMgdGhlIGJvZHksIHdoaWNoIGlzIHRoZSBjb21tYW5kcyB0aGF0IHdpbGwgYmUgZXhlY3V0ZWQgKmlmKiB0aGUgCnRlc3QgaXMgYFRSVUVgLgpUaGVzZSBhcmUgcGxhY2VkIHdpdGhpbiBhIHNldCBvZiBicmFjZXMgYHsgfWAuCk5vdGUgdGhhdCB3ZSB1c2VkIGFuIGV4Y2xhbWF0aW9uIHBvaW50IGluIHRoZSB0ZXN0IHRvIHNpZ25pZnkgdGhhdCB3ZSB3YW50IGEgCmRpcmVjdG9yeSB0byBiZSBjcmVhdGVkIG9ubHkgKmlmKiBgZGlyLmV4aXN0cyhyZXN1bHRzKWAgaXMgTk9UIGVxdWFsIHRvIGBUUlVFYC4KCmBgYHtyfQojIElmICdyZXN1bHRzJyBkaXJlY3RvcnkgZG9lc24ndCBleGlzdC4uLgppZiAoIWRpci5leGlzdHMoInJlc3VsdHMiKSkgewogICMgLi4uIGNyZWF0ZSBhICdyZXN1bHRzJyBkaXJlY3RvcnkKICBkaXIuY3JlYXRlKCJyZXN1bHRzIikKfQpgYGAKClRoZSBgZGlyLmV4aXN0cygpYCBmdW5jdGlvbiB3aWxsIG5vdCB3b3JrIG9uIGZpbGVzIHRoZW1zZWx2ZXMuCkluIHRoYXQgY2FzZSwgdGhlcmUgaXMgYW4gYW5hbG9nb3VzIGZ1bmN0aW9uIGNhbGxlZCBgZmlsZS5leGlzdHMoKWAuCgpUcnkgdXNpbmcgdGhlIGBmaWxlLmV4aXN0cygpYCBmdW5jdGlvbiB0byBzZWUgaWYgdGhlIGZpbGUgCmBnZW5lX3Jlc3VsdHNfR1NFNDQ5NzEudHN2YCBleGlzdHMgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5LgpVc2UgdGhlIGNvZGUgY2h1bmsgd2Ugc2V0IHVwIGZvciB5b3UgYmVsb3cuIApOb3RlIHRoYXQgaW4gb3VyIG5vdGVib29rcyAoYW5kIHNvbWV0aW1lcyBlbHNld2hlcmUpLCB3aGVyZXZlciB5b3Ugc2VlIGEgCmA8RklMTF9JTl9USEVfQkxBTks+YCBsaWtlIGluIHRoZSBjaHVuayBiZWxvdywgdGhhdCBpcyBtZWFudCBmb3IgeW91IHRvIHJlcGxhY2UgCihpbmNsdWRpbmcgdGhlIGFuZ2xlIGJyYWNrZXRzKSB3aXRoIHRoZSBjb3JyZWN0IHBocmFzZSBiZWZvcmUgeW91IHJ1biB0aGUgY2h1bmsgCihvdGhlcndpc2UgeW91IHdpbGwgZ2V0IGFuIGVycm9yKS4KCmBgYHtyIGV2YWw9RkFMU0V9CiMgUmVwbGFjZSB0aGUgPFBVVF9GSUxFX05BTUVfSEVSRT4gd2l0aCB0aGUgbmFtZSBvZiB0aGUgZmlsZSB5b3UgYXJlIGxvb2tpbmcgZm9yCiMgUmVtZW1iZXIgdG8gdXNlIHF1b3RlcyB0byBtYWtlIGl0IGEgY2hhcmFjdGVyIHN0cmluZwpmaWxlLmV4aXN0cyg8UFVUX0ZJTEVfTkFNRV9IRVJFPikKYGBgCgpOb3cgdGhhdCB3ZSd2ZSBkZXRlcm1pbmVkIHRoYXQgYGdlbmVfcmVzdWx0c19HU0U0NDk3MS50c3ZgIGV4aXN0cywgd2UgYXJlIHJlYWR5IAp0byByZWFkIGl0IGludG8gb3VyIFIgZW52aXJvbm1lbnQuCgojIyMjIFJlYWQgYSBUU1YgZmlsZQoKRGVjbGFyZSB0aGUgbmFtZSBvZiB0aGUgZGlyZWN0b3J5IHdoZXJlIHdlIHdpbGwgcmVhZCBpbiB0aGUgZGF0YS4gCgpgYGB7cn0KZGF0YV9kaXIgPC0gImRhdGEiCmBgYAoKQWx0aG91Z2ggYmFzZSBSIGhhcyBmdW5jdGlvbnMgdG8gcmVhZCBpbiBkYXRhIGZpbGVzLCB0aGUgZnVuY3Rpb25zIGluIHRoZSAKYHJlYWRyYCBwYWNrYWdlIChwYXJ0IG9mIHRoZSB0aWR5dmVyc2UpIGFyZSBmYXN0ZXIgYW5kIG1vcmUgc3RyYWlnaHRmb3J3YXJkIAp0byB1c2Ugc28gd2UgYXJlIGdvaW5nIHRvIHVzZSB0aG9zZSBoZXJlLiAKQmVjYXVzZSB0aGUgZmlsZSB3ZSBhcmUgcmVhZGluZyBpbiBpcyBhIFRTViAodGFiIHNlcGFyYXRlZCB2YWx1ZXMpIGZpbGUgd2Ugd2lsbCAKYmUgdXNpbmcgdGhlIGByZWFkX3RzdmAgZnVuY3Rpb24uIApUaGVyZSBhcmUgYW5hbG9nb3VzIGZ1bmN0aW9ucyBmb3IgQ1NWIChjb21tYSBzZXBhcmF0ZWQgdmFsdWVzKSBmaWxlcyAKKGByZWFkX2NzdigpYCkgYW5kIG90aGVyIGZpbGVzIHR5cGVzLgoKIyMgUmVhZCBpbiB0aGUgZGlmZmVyZW50aWFsIGV4cHJlc3Npb24gYW5hbHlzaXMgcmVzdWx0cyBmaWxlCgpgYGB7cn0Kc3RhdHNfZGYgPC0gcmVhZHI6OnJlYWRfdHN2KGZpbGUucGF0aChkYXRhX2RpciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiZ2VuZV9yZXN1bHRzX0dTRTQ0OTcxLnRzdiIpKQpgYGAKCkZvbGxvd2luZyB0aGUgdGVtcGxhdGUgb2YgdGhlIHByZXZpb3VzIGNodW5rLCB1c2UgdGhpcyBjaHVuayB0byByZWFkIGluIHRoZSBmaWxlCmBHU0U0NDk3MS50c3ZgIHRoYXQgaXMgaW4gdGhlIGBkYXRhYCBmb2xkZXIuIAoKYGBge3J9CiMgVXNlIHRoaXMgY2h1bmsgdG8gcmVhZCBpbiBkYXRhIGZyb20gdGhlIGZpbGUgYEdTRTQ0OTcxLnRzdmAKYGBgCgpVc2UgdGhpcyBjaHVuayB0byBleHBsb3JlIHdoYXQgYGdlbmVfZGZgIGxvb2tzIGxpa2UuIAoKYGBge3J9CiMgRXhwbG9yZSBgZ2VuZV9kZmAKYGBgCgpXaGF0IGluZm9ybWF0aW9uIGlzIGNvbnRhaW5lZCBpbiBgZ2VuZV9kZmA/CgojIyBkcGx5ciBwaXBlcwoKT25lIG5pZnR5IGZlYXR1cmUgb2YgdGhlIHRpZHl2ZXJzZSBpcyBwaXBlczogYCU+JWAKVGhlc2UgaGFuZHkgdGhpbmdzIGFsbG93cyB5b3UgdG8gZnVubmVsIHRoZSByZXN1bHQgb2Ygb25lIGV4cHJlc3Npb24gdG8gdGhlIG5leHQsCm1ha2luZyB5b3VyIGNvZGUgYSBsaXR0bGUgbW9yZSBzdHJlYW1saW5lZC4KCkZvciBleGFtcGxlLCB0aGUgb3V0cHV0IGZyb20gdGhpczogIAoKYGBge3J9CmZpbHRlcihzdGF0c19kZiwgY29udHJhc3QgPT0gIm1hbGVfZmVtYWxlIikKYGBgICAKCi4uLmlzIHRoZSBzYW1lIGFzIHRoZSBvdXRwdXQgZnJvbSB0aGlzOiAgCgpgYGB7cn0Kc3RhdHNfZGYgJT4lIGZpbHRlcihjb250cmFzdCA9PSAibWFsZV9mZW1hbGUiKQpgYGAgIAogIApUaGlzIGNhbiBtYWtlIHlvdXIgY29kZSBjbGVhbmVyIGFuZCBlYXNpZXIgdG8gZm9sbG93IGEgc2VyaWVzIG9mIHJlbGF0ZWQgCmNvbW1hbmRzLiAKTGV0J3MgbG9vayBhdCBhbiBleGFtcGxlIHdpdGggb3VyIHN0YXRzIG9mIG9mIGhvdyB0aGUgc2FtZSAKZnVuY3Rpb25zIGxvb2sgd2l0aCBvciB3aXRob3V0IHBpcGVzOgoKKkV4YW1wbGUgMToqIHdpdGhvdXQgcGlwZXM6IAoKYGBge3J9CnN0YXRzX2FycmFuZ2VkIDwtIGFycmFuZ2Uoc3RhdHNfZGYsIHRfc3RhdGlzdGljKQpzdGF0c19maWx0ZXJlZCA8LSBmaWx0ZXIoc3RhdHNfYXJyYW5nZWQsIGF2Z19leHByZXNzaW9uID4gNTApCnN0YXRzX25vcGlwZSA8LSBzZWxlY3Qoc3RhdHNfZmlsdGVyZWQsIGNvbnRyYXN0LCBsb2dfZm9sZF9jaGFuZ2UsIHBfdmFsdWUpCmBgYAogIApVR0gsIHdlIGhhdmUgdG8ga2VlcCB0cmFjayBvZiBhbGwgb2YgdGhvc2UgZGlmZmVyZW50IGludGVybWVkaWF0ZSBkYXRhIGZyYW1lcyAKYW5kIHR5cGUgdGhlaXIgbmFtZXMgc28gbWFueSB0aW1lcyBoZXJlISAKV2UgY291bGQgbWF5YmUgc3RyZWFtbGluZSB0aGluZ3MgYnkgdXNpbmcgdGhlIHNhbWUgdmFyaWFibGUgbmFtZSBhdCBlYWNoIHN0YWdlLCAKYnV0IGV2ZW4gdGhlbiB0aGVyZSBpcyBhIGxvdCBvZiBleHRyYSB0eXBpbmcsIGFuZCBpdCBpcyBlYXN5IHRvIGdldCBjb25mdXNlZCAKYWJvdXQgd2hhdCBoYXMgYmVlbiBkb25lIHdoZXJlLgpJdCdzIGFubm95aW5nIGFuZCBtYWtlcyBpdCBoYXJkZXIgZm9yIHBlb3BsZSB0byByZWFkLiAKICAKKkV4YW1wbGUgMjoqIFNhbWUgcmVzdWx0IGFzIDEgYnV0IHdpdGggcGlwZXMhCgpgYGB7cn0KIyBFeGFtcGxlIG9mIHRoZSBzYW1lIG1vZGlmaWNhdGlvbnMgYXMgYWJvdmUgYnV0IHdpdGggcGlwZXMhCnN0YXRzX3BpcGUgIDwtIHN0YXRzX2RmICU+JQogICAgICAgICAgICAgICBhcnJhbmdlKHRfc3RhdGlzdGljKSAlPiUKICAgICAgICAgICAgICAgZmlsdGVyKGF2Z19leHByZXNzaW9uID4gNTApICU+JQogICAgICAgICAgICAgICBzZWxlY3QoY29udHJhc3QsIGxvZ19mb2xkX2NoYW5nZSwgcF92YWx1ZSkKYGBgCgpXaGF0IHRoZSBgJT4lYCAocGlwZSkgaXMgZG9pbmcgaGVyZSBpcyBmZWVkaW5nIHRoZSByZXN1bHQgb2YgdGhlIGV4cHJlc3Npb24gb24gCml0cyBsZWZ0IGludG8gdGhlIGZpcnN0IGFyZ3VtZW50IG9mIHRoZSBuZXh0IGZ1bmN0aW9uICh0byBpdHMgcmlnaHQsIG9yIG9uIHRoZSAKbmV4dCBsaW5lIGhlcmUpLiAKV2UgY2FuIHRoZW4gc2tpcCB0aGF0IGZpcnN0IGFyZ3VtZW50ICh0aGUgZGF0YSBpbiB0aGVzZSBjYXNlcyksIGFuZCBtb3ZlIHJpZ2h0IApvbiB0byB0aGUgcGFydCB3ZSBjYXJlIGFib3V0IGF0IHRoYXQgc3RlcDogd2hhdCB3ZSBhcmUgYXJyYW5naW5nLCBmaWx0ZXJpbmcsIG9yIApzZWxlY3RpbmcgIGluIHRoaXMgY2FzZS4KCkxldCdzIGRvdWJsZSBjaGVjayB0aGF0IHRoZXNlIGFyZSB0aGUgc2FtZSBieSB1c2luZyB0aGUgZnVuY3Rpb24sIGBhbGwuZXF1YWwoKWAuIAoKYGBge3J9CmFsbC5lcXVhbChzdGF0c19ub3BpcGUsIHN0YXRzX3BpcGUpCmBgYAoKYGFsbC5lcXVhbCgpYCBpcyBsZXR0aW5nIHVzIGtub3cgdGhhdCB0aGVzZSB0d28gb2JqZWN0cyBhcmUgdGhlIHNhbWUuIAoKTm93IHRoYXQgaG9wZWZ1bGx5IHlvdSBhcmUgY29udmluY2VkIHRoYXQgdGhlIHRpZHl2ZXJzZSBjYW4gaGVscCB5b3UgbWFrZSB5b3VyIApjb2RlIG5lYXRlciBhbmQgZWFzaWVyIHRvIHVzZSBhbmQgcmVhZCwgbGV0J3MgZ28gdGhyb3VnaCBzb21lIG9mIHRoZSBwb3B1bGFyIAp0aWR5dmVyc2UgZnVuY3Rpb25zIGFuZCBzbyB3ZSBjYW4gY3JlYXRlIHBpcGVsaW5lcyBsaWtlIHRoaXMuIAoKIyMgQ29tbW9uIHRpZHl2ZXJzZSBmdW5jdGlvbnMKCkxldCdzIHNheSB3ZSB3YW50ZWQgdG8gZmlsdGVyIHRoaXMgZ2VuZSBleHByZXNzaW9uIGRhdGFzZXQgdG8gcGFydGljdWxhciBzYW1wbGUKZ3JvdXBzLgpJbiBvcmRlciB0byBkbyB0aGlzLCB3ZSB3b3VsZCB1c2UgdGhlIGZ1bmN0aW9uIGBmaWx0ZXIoKWAgYXMgd2VsbCBhcyBhIGxvZ2ljIApzdGF0ZW1lbnQgKHVzdWFsbHkgb25lIHRoYXQgcmVmZXJzIHRvIGEgY29sdW1uIG9yIGNvbHVtbnMgaW4gdGhlIGRhdGEgZnJhbWUpLgoKYGBge3J9CiMgSGVyZSBsZXQncyBmaWx0ZXIgdGhlIGRhdGEgdG8gYSBwYXJ0aWN1bGFyIGdlbmUKc3RhdHNfZGYgJT4lIAogIGZpbHRlcihnZW5lX3N5bWJvbCA9PSAiU05DQSIpCmBgYAoKV2UgY2FuIHVzZSBgZmlsdGVyKClgIHNpbWlsYXJseSBmb3IgbnVtZXJpYyBzdGF0ZW1lbnRzLiAgCgpgYGB7cn0KIyBIZXJlIGxldCdzIGZpbHRlciB0aGUgZGF0YSB0byBoYXZlIGF2ZXJhZ2UgZXhwcmVzc2lvbiB2YWx1ZXMgYWJvdmUgNTAKc3RhdHNfZGYgJT4lIAogIGZpbHRlcihhdmdfZXhwcmVzc2lvbiA+IDUwKQpgYGAKCldlIGNhbiBhcHBseSBtdWx0aXBsZSBmaWx0ZXJzIGF0IG9uY2UsIHdoaWNoIHdpbGwgcmVxdWlyZSBhbGwgb2YgdGhlbSB0byBiZSAKc2F0aXNmaWVkIGZvciBldmVyeSByb3cgaW4gdGhlIHJlc3VsdHM6CgpgYGB7ciB9CnN0YXRzX2RmICU+JSAKICBmaWx0ZXIoY29udHJhc3QgPT0gIm1hbGVfZmVtYWxlIiwgCiAgICAgICAgIGF2Z19leHByZXNzaW9uID4gNTApCmBgYAoKV2hlbiB3ZSBhcmUgZmlsdGVyaW5nLCB0aGUgYCVpbiVgIG9wZXJhdG9yIGNhbiBjb21lIGluIGhhbmR5IGlmIHdlIGhhdmUgbXVsdGlwbGUKaXRlbXMgd2Ugd291bGQgbGlrZSB0byBtYXRjaC4KTGV0J3MgdGFrZSBhIGxvb2sgYXQgd2hhdCB1c2luZyBgJWluJWAgZG9lcy4KCmBgYHtyfQpzdGF0c19kZiRnZW5lX3N5bWJvbCAlaW4lIGMoIlNOQ0EiLCAiQ0RLTjFBIikKYGBgCgpgJWluJWAgcmV0dXJucyBhIGxvZ2ljYWwgdmVjdG9yIHRoYXQgbm93IHdlIGNhbiB1c2UgaW4gYGRwbHlyOjpmaWx0ZXJgLgoKYGBge3IgfQpzdGF0c19kZiAlPiUgCiAgZmlsdGVyKGdlbmVfc3ltYm9sICVpbiUgYygiU05DQSIsICJDREtOMUEiKSkKYGBgCgpMZXQncyByZXR1cm4gdG8gb3VyIGZpcnN0IGBmaWx0ZXIoKWAgYW5kIGJ1aWxkIG9uIHRvIGl0LiAKVGhpcyB0aW1lLCBsZXQncyBrZWVwIG9ubHkgc29tZSBvZiB0aGUgY29sdW1ucyBmcm9tIHRoZSBkYXRhIGZyYW1lIHVzaW5nIHRoZSAKYHNlbGVjdCgpYCBmdW5jdGlvbi4gCkxldCdzIGFsc28gc2F2ZSB0aGlzIGFzIGEgbmV3IGRhdGEgZnJhbWUgY2FsbGVkIGBzdGF0c19maWx0ZXJlZF9kZmAuCgpgYGB7cn0Kc3RhdHNfZmlsdGVyZWRfZGYgPC0gc3RhdHNfZGYgJT4lIAogIGZpbHRlcihjb250cmFzdCA9PSAibWFsZV9mZW1hbGUiLCAKICAgICAgICAgYXZnX2V4cHJlc3Npb24gPiA1MCkgJT4lCiAgc2VsZWN0KGxvZ19mb2xkX2NoYW5nZSwgdF9zdGF0aXN0aWMpCmBgYAoKTGV0J3Mgc2F5IHdlIHdhbnRlZCB0byBhcnJhbmdlIHRoaXMgZGF0YXNldCBzbyB0aGF0IHRoZSBnZW5lcyBhcmUgYXJyYW5nZWQgYnkgCnRoZSBzbWFsbGVzdCBwIHZhbHVlcyB0byB0aGUgbGFyZ2VzdC4KSW4gb3JkZXIgdG8gZG8gdGhpcywgd2Ugd291bGQgdXNlIHRoZSBmdW5jdGlvbiBgYXJyYW5nZSgpYCBhcyB3ZWxsIGFzIHRoZSBjb2x1bW4Kd2Ugd291bGQgbGlrZSB0byBzb3J0IGJ5IChpbiB0aGlzIGNhc2UgYHBfdmFsdWVgKS4KCmBgYHtyfQpzdGF0c19kZiAlPiUgCiAgYXJyYW5nZShwX3ZhbHVlKSAKYGBgCgpXaGF0IGlmIHdlIHdhbnQgdG8gc29ydCBmcm9tIGxhcmdlc3QgdG8gc21hbGxlc3Q/IApMaWtlIGlmIHdlIHdhbnQgdG8gc2VlIHRoZSBnZW5lcyB3aXRoIHRoZSBoaWdoZXN0IGF2ZXJhZ2UgZXhwcmVzc2lvbj8KV2UgY2FuIHVzZSB0aGUgc2FtZSBmdW5jdGlvbiwgYnV0IGluc3RlYWQgdXNlIHRoZSBgZGVzYygpYCBmdW5jdGlvbiBhbmQgbm93IHdlIAphcmUgdXNpbmcgYGF2Z19leHByZXNzaW9uYCBjb2x1bW4uIAoKYGBge3J9CnN0YXRzX2RmICU+JSAKICBhcnJhbmdlKGRlc2MoYXZnX2V4cHJlc3Npb24pKQpgYGAgCgpXaGF0IGlmIHdlIHdvdWxkIGxpa2UgdG8gY3JlYXRlIGEgbmV3IGNvbHVtbiBvZiB2YWx1ZXM/CkZvciB0aGF0IHdlIHVzZSBgbXV0YXRlKClgIGZ1bmN0aW9uLgoKYGBge3J9CnN0YXRzX2RmICU+JSAKICBtdXRhdGUobG9nMTBfcF92YWx1ZSA9IC1sb2cxMChwX3ZhbHVlKSkKYGBgCgpXaGF0IGlmIHdlIHdhbnQgdG8gb2J0YWluIHN1bW1hcnkgc3RhdGlzdGljcyBmb3IgYSBjb2x1bW4/IApUaGUgYHN1bW1hcml6ZWAgZnVuY3Rpb24gYWxsb3dzIHVzIHRvIGNhbGN1bGF0ZSBzdW1tYXJ5IHN0YXRpc3RpY3MgZm9yIGEgY29sdW1uLiAKSGVyZSB3ZSB3aWxsIHVzZSBzdW1tYXJpemUgdG8gb2J0YWluIGFuIG1lYW4gbG9nIGZvbGRlciBjaGFuZ2Ugb3ZlciBhbGwgdGhlIApnZW5lcy4gCgpgYGB7cn0Kc3RhdHNfZGYgJT4lIAogIHN1bW1hcml6ZShtZWFuKGxvZ19mb2xkX2NoYW5nZSkpCmBgYAoKV2hhdCBpZiB3ZSdkIGxpa2UgdG8gb2J0YWluIGEgc3VtbWFyeSBzdGF0aXN0aWNzIGJ1dCBoYXZlIHRoZW0gZm9yIHZhcmlvdXMgCmdyb3Vwcz8KQ29udmVuaWVudGx5IG5hbWVkLCB0aGVyZSdzIGEgZnVuY3Rpb24gY2FsbGVkIGBncm91cF9ieSgpYCB0aGF0IHNlYW1sZXNzbHkgCmFsbG93cyB1cyB0byBkbyB0aGlzLiAKQWxzbyBub3RlIHRoYXQgYGdyb3VwX2J5KClgIGFsbG93cyB1cyB0byBncm91cCBieSBtdWx0aXBsZSB2YXJpYWJsZXMgYXQgYSB0aW1lIAppZiB5b3Ugd2FudCB0by4KCmBgYHtyfQpzdGF0c19zdW1tYXJ5X2RmIDwtIHN0YXRzX2RmICU+JQogICAgICBncm91cF9ieShjb250cmFzdCkgJT4lIAogICAgICBzdW1tYXJpemUobWVhbihsb2dfZm9sZF9jaGFuZ2UpKQpgYGAKCkxldCdzIGxvb2sgYXQgYSBwcmV2aWV3IG9mIHdoYXQgd2UgbWFkZToKCmBgYHtyfQpzdGF0c19zdW1tYXJ5X2RmCmBgYAoKSGVyZSB3ZSBoYXZlIHRoZSBtZWFuIGxvZyBmb2xkIGNoYW5nZSBleHByZXNzaW9uIHBlciBlYWNoIGNvbnRyYXN0IHdlIG1hZGUuIAoKIyMgQSBicmllZiBpbnRybyB0byB0aGUgYGFwcGx5YCBmYW1pbHkgb2YgZnVuY3Rpb25zCgpJbiBiYXNlIFIsIHRoZSBgYXBwbHlgIGZhbWlseSBvZiBmdW5jdGlvbnMgY2FuIGJlIGFuIGFsdGVybmF0aXZlIG1ldGhvZHMgZm9yIApwZXJmb3JtaW5nIHRyYW5zZm9ybWF0aW9ucyBhY3Jvc3MgYSBkYXRhIGZyYW1lLCBtYXRyaXggb3Igb3RoZXIgb2JqZWN0IHN0cnVjdHVyZXMuIAoKT25lIG9mIHRoaXMgZmFtaWx5IGlzIChzaG9ja2luZ2x5KSB0aGUgZnVuY3Rpb24gYGFwcGx5KClgLCB3aGljaCBvcGVyYXRlcyBvbiAKbWF0cmljZXMgb3IgZGF0YSBmcmFtZXMuIApUaGUgZmlyc3QgYXJndW1lbnQgdG8gYGFwcGx5KClgIGlzIHRoZSBkYXRhIG9iamVjdCB3ZSB3YW50IHRvIHdvcmsgb24uClRoZSB0aGlyZCBhcmd1bWVudCBpcyB0aGUgZnVuY3Rpb24gd2Ugd2lsbCBhcHBseSB0byBlYWNoIHJvdyBvciBjb2x1bW4gb2YgdGhlIApkYXRhIG9iamVjdC4KVGhlIHNlY29uZCBhcmd1bWVudCBpbiBzcGVjaWZpZXMgd2hldGhlciB3ZSBhcmUgYXBwbHlpbmcgdGhlIGZ1bmN0aW9uIAphY3Jvc3Mgcm93cyBvciBhY3Jvc3MgY29sdW1ucyAoMSBmb3Igcm93cywgMiBmb3IgY29sdW1ucykuCgpSZW1lbWJlciB0aGF0IGBnZW5lX2RmYCBpcyBhIGdlbmUgeCBzYW1wbGUgZ2VuZSBleHByZXNzaW9uIGRhdGEgZnJhbWUuIAoKYGBge3J9CiMgQ2FsY3VsYXRlIHJvdyBtZWFucwpnZW5lX21lYW5zIDwtIGFwcGx5KGdlbmVfZGZbLCAtMV0sIDEsIG1lYW4pICMgTm90aWNlIHdlIGFyZSB1c2luZyAxIGhlcmUKCiMgSG93IGxvbmcgd2lsbCBgZ2VuZV9tZWFuc2AgYmU/IApsZW5ndGgoZ2VuZV9tZWFucykKYGBgCgpXaHkgZG8gd2UgaGF2ZSBhIGBbLCAtMV1gIGFmdGVyIGBnZW5lX2RmYCBpbiB0aGUgYWJvdmUgY2h1bms/CgpOb3cgbGV0J3MgaW52ZXN0aWdhdGUgdGhlIHNhbWUgc2V0IHVwLCBidXQgdXNlIDIgdG8gYGFwcGx5YCBvdmVyIHRoZSBjb2x1bW5zIG9mCm91ciBtYXRyaXguCgpgYGB7cn0KIyBDYWxjdWxhdGUgc2FtcGxlIG1lYW5zCnNhbXBsZV9tZWFucyA8LSBhcHBseShnZW5lX2RmWywgLTFdLCAyLCBtZWFuKSAjIE5vdGljZSB3ZSB1c2UgMiBoZXJlCgojIEhvdyBsb25nIHdpbGwgYHNhbXBsZV9tZWFuc2AgYmU/IApsZW5ndGgoc2FtcGxlX21lYW5zKQpgYGAKCkFsdGhvdWdoIHRoZSBgYXBwbHlgIGZ1bmN0aW9ucyBtYXkgbm90IGJlIGFzIGVhc3kgdG8gdXNlIGFzIHRoZSB0aWR5dmVyc2UgCmZ1bmN0aW9ucywgZm9yIHNvbWUgYXBwbGljYXRpb25zLCBgYXBwbHlgIG1ldGhvZHMgY2FuIGJlIGJldHRlciBzdWl0ZWQuCkluIHRoaXMgd29ya3Nob3AsIHdlIHdpbGwgbm90IGRlbHZlIHRvbyBkZWVwbHkgaW50byB0aGUgdmFyaW91cyBvdGhlciBhcHBseSAKZnVuY3Rpb25zIChgdGFwcGx5KClgLCBgbGFwcGx5KClgLCBldGMuKSBidXQgeW91IGNhbiByZWFkIG1vcmUgZXhhbXBsZXMgYWJvdXQgCnRoZW0gW2hlcmVdKGh0dHBzOi8vd3d3Lmd1cnU5OS5jb20vci1hcHBseS1zYXBwbHktdGFwcGx5Lmh0bWwpLgoKIyMgVGhlIGRwbHlyOjpqb2luIGZ1bmN0aW9ucwoKTGV0J3Mgc2F5IHdlIGhhdmUgYSBzY2VuYXJpbyB3aGVyZSB3ZSBoYXZlIHR3byBkYXRhIGZyYW1lcyB0aGF0IHdlIHdvdWxkIGxpa2UgdG8gCmNvbWJpbmUuIApSZWNhbGwgdGhhdCBgc3RhdHNfZGZgIGFuZCBgZ2VuZV9kZmAgYXJlIGRhdGEgZnJhbWVzIHRoYXQgY29udGFpbiBpbmZvcm1hdGlvbiAKYWJvdXQgc29tZSBvZiB0aGUgc2FtZSBnZW5lcy4KVGhlIFtgZHBseXI6OmpvaW5gIGZhbWlseSBvZiBmdW5jdGlvbnNdKGh0dHBzOi8vZHBseXIudGlkeXZlcnNlLm9yZy9yZWZlcmVuY2Uvam9pbi5odG1sKSAKYXJlIHVzZWZ1bCBmb3IgdmFyaW91cyBzY2VuYXJpb3Mgb2YgY29tYmluaW5nIGRhdGEgZnJhbWVzLiAKCkZvciBub3csIHdlIHdpbGwgZm9jdXMgb24gYGlubmVyX2pvaW4oKWAsIHdoaWNoIHdpbGwgY29tYmluZSBkYXRhIGZyYW1lcyBieSBvbmx5CmtlZXBpbmcgaW5mb3JtYXRpb24gYWJvdXQgbWF0Y2hpbmcgcm93cyB0aGF0IGFyZSBpbiBib3RoIGRhdGEgZnJhbWVzLgpXZSBuZWVkIHRvIHVzZSB0aGUgYGJ5YCBhcmd1bWVudCB0byBkZXNpZ25hdGUgd2hhdCBjb2x1bW4ocykgCnNob3VsZCBiZSB1c2VkIGFzIGEga2V5IHRvIG1hdGNoIHRoZSBkYXRhIGZyYW1lcy4KSW4gdGhpcyBjYXNlIHdlIHdhbnQgdG8gbWF0Y2ggdGhlIGdlbmUgaW5mb3JtYXRpb24gYmV0d2VlbiB0aGUgdHdvLCBzbyB3ZSB3aWxsIApzcGVjaWZ5IHRoYXQgd2Ugd2FudCB0byBjb21wYXJlIHZhbHVlcyBpbiB0aGUgYGVuc2VtYmxfaWRgIGNvbHVtbiBmcm9tIApgc3RhdHNfZGZgIHRvIHRoZSBgR2VuZWAgY29sdW1uIGZyb20gYGdlbmVfZGZgLgoKYGBge3J9CnN0YXRzX2RmICU+JSAKICBpbm5lcl9qb2luKGdlbmVfZGYsIGJ5ID0gYygnZW5zZW1ibF9pZCcgPSAnR2VuZScpKSAKYGBgCgojIyBTYXZlIGRhdGEgdG8gZmlsZXMKCiMjIyMgU2F2ZSB0byBUU1YgZmlsZXMKCkxldCdzIHdyaXRlIHNvbWUgb2YgdGhlIGRhdGEgZnJhbWVzIHdlIGNyZWF0ZWQgdG8gYSBmaWxlLgpUbyBkbyB0aGlzLCB3ZSBjYW4gdXNlIHRoZSBgcmVhZHJgIGxpYnJhcnkgb2YgYF93cml0ZSgpYCBmdW5jdGlvbnMuIApUaGUgZmlyc3QgYXJndW1lbnQgb2YgYHdyaXRlX3RzdigpYCBpcyB0aGUgZGF0YSB3ZSB3YW50IHRvIHdyaXRlLCBhbmQgdGhlIHNlY29uZCAKYXJndW1lbnQgaXMgYSBjaGFyYWN0ZXIgc3RyaW5nIHRoYXQgZGVzY3JpYmVzIHRoZSBwYXRoIHRvIHRoZSBuZXcgZmlsZSB3ZSB3b3VsZCAKbGlrZSB0byBjcmVhdGUuClJlbWVtYmVyIHRoYXQgd2UgY3JlYXRlZCBhIGByZXN1bHRzYCBkaXJlY3RvcnkgdG8gcHV0IG91ciBvdXRwdXQgaW4sIApidXQgaWYgd2Ugd2FudCB0byBzYXZlIG91ciBkYXRhIHRvIGEgZGlyZWN0b3J5IG90aGVyIHRoYW4gb3VyIHdvcmtpbmcgZGlyZWN0b3J5LCAKd2UgbmVlZCB0byBzcGVjaWZ5IHRoaXMuIApUaGlzIGlzIHdoYXQgd2Ugd2lsbCB1c2UgdGhlIGBmaWxlLnBhdGgoKWAgZnVuY3Rpb24gZm9yLiAKTGV0J3MgbG9vayBpbiBhIGJpdCBtb3JlIGRldGFpbCB3aGF0IGBmaWxlLnBhdGgoKWAgZG9lcywgYnkgZXhhbWluaW5nIHRoZSAKcmVzdWx0cyBvZiB0aGUgZnVuY3Rpb24gaW4gdGhlIGV4YW1wbGVzIGJlbG93LgoKYGBge3J9CiMgV2hpY2ggb2YgdGhlc2UgZmlsZSBwYXRocyBpcyB3aGF0IHdlIHdhbnQgdG8gdXNlIHRvIHNhdmUgb3VyIGRhdGEgdG8gdGhlCiMgcmVzdWx0cyBkaXJlY3Rvcnkgd2UgY3JlYXRlZCBhdCB0aGUgYmVnaW5uaW5nIG9mIHRoaXMgbm90ZWJvb2s/CmZpbGUucGF0aCgiZG9ja2VyLWluc3RhbGwiLCAic3RhdHNfc3VtbWFyeS50c3YiKQpmaWxlLnBhdGgoInJlc3VsdHMiLCAic3RhdHNfc3VtbWFyeS50c3YiKQpmaWxlLnBhdGgoInN0YXRzX3N1bW1hcnkudHN2IiwgInJlc3VsdHMiKQpgYGAKClJlcGxhY2UgYDxORVdfRklMRV9QQVRIPmAgYmVsb3cgd2l0aCB0aGUgYGZpbGUucGF0aCgpYCBzdGF0ZW1lbnQgZnJvbSBhYm92ZSB0aGF0IAp3aWxsIHN1Y2Nlc3NmdWxseSBzYXZlIG91ciBmaWxlIHRvIHRoZSBgcmVzdWx0c2AgZm9sZGVyIAoKYGBge3IgZXZhbD1GQUxTRX0KIyBXcml0ZSBvdXIgZGF0YSBmcmFtZSB0byBhIFRTViBmaWxlCnJlYWRyOjp3cml0ZV90c3Yoc3RhdHNfc3VtbWFyeV9kZiwgPE5FV19GSUxFX1BBVEg+KQpgYGAKCkNoZWNrIGluIHlvdXIgYHJlc3VsdHNgIGRpcmVjdG9yeSB0byBzZWUgaWYgeW91ciBuZXcgZmlsZSBoYXMgc3VjY2Vzc2Z1bGx5IHNhdmVkLgoKIyMjIyBTYXZlIHRvIFJEUyBmaWxlcwoKRm9yIHRoaXMgZXhhbXBsZSB3ZSBoYXZlIGJlZW4gd29ya2luZyB3aXRoIGRhdGEgZnJhbWVzLCB3aGljaCBhcmUgY29udmVuaWVudGx5IApyZXByZXNlbnRlZCBhcyBUU1Ygb3IgQ1NWIHRhYmxlcy4gCkhvd2V2ZXIsIGluIG90aGVyIHNpdHVhdGlvbnMgd2UgbWF5IHdhbnQgdG8gc2F2ZSBtb3JlIGNvbXBsaWNhdGVkIG9yIHZlcnkgbGFyZ2UgCmRhdGEgc3RydWN0dXJlcywgUkRTIChSIERhdGEgU2VyaWFsaXplZC9TaW5nbGUpIGZpbGVzIG1heSBiZSBhIGJldHRlciBvcHRpb24gZm9yCnNhdmluZyBvdXIgZGF0YS4KUkRTIGlzIFIncyBzcGVjaWFsIGZpbGUgZm9ybWF0IGZvciBob2xkaW5nIGRhdGEgZXhhY3RseSBhcyB5b3UgaGF2ZSBpdCBpbiB5b3VyIApSIGVudmlyb25tZW50LiAKUkRTIGZpbGVzIGNhbiBhbHNvIGJlIGNvbXByZXNzZWQsIG1lYW5pbmcgdGhleSB3aWxsIHRha2UgdXAgbGVzcyBzcGFjZSBvbiB5b3VyIApjb21wdXRlci4gCkxldCdzIHNhdmUgb3VyIGRhdGEgdG8gYW4gUkRTIGZpbGUgaW4gb3VyIGByZXN1bHRzYCBmb2xkZXIuCllvdSB3aWxsIG5lZWQgdG8gcmVwbGFjZSB0aGUgYC50c3ZgIHdpdGggYC5SRFNgLCBidXQgeW91IGNhbiB1c2Ugd2hhdCB3ZSAKZGV0ZXJtaW5lZCBhcyBvdXIgZmlsZSBwYXRoIGZvciB0aGUgbGFzdCBjaHVuayBhcyB5b3VyIHRlbXBsYXRlLiAKCmBgYHtyIGV2YWw9RkFMU0V9CiMgV3JpdGUgeW91ciBvYmplY3QgdG8gYW4gUkRTIGZpbGUKcmVhZHI6OndyaXRlX3JkcyhzdGF0c19zdW1tYXJ5LCA8UFVUX0NPUlJFQ1RfRklMRV9QQVRIX0hFUkU+KQpgYGAKCiMjIyMgUmVhZCBhbiBSRFMgZmlsZQoKU2luY2Ugbm93IHlvdSBoYXZlIGxlYXJuZWQgdGhlIGByZWFkcmAgZnVuY3Rpb25zOiBgcmVhZF90c3YoKWAsIGB3cml0ZV90c3YoKWAsIAphbmQgbm93LCBgd3JpdGVfcmRzKClgLCB3aGF0IGRvIHlvdSBzdXBwb3NlIHRoZSBmdW5jdGlvbiB5b3Ugd2lsbCBuZWVkIHRvIHJlYWQgCnlvdXIgUkRTIGZpbGUgaXMgY2FsbGVkPyAKVXNlIHRoYXQgZnVuY3Rpb24gaGVyZSB0byByZS1pbXBvcnQgeW91ciBkYXRhIGluIHRoZSBjaHVuayB3ZSBzZXQgdXAgZm9yIHlvdQpiZWxvdy4KCmBgYHtyIGV2YWw9RkFMU0V9CiMgUmVhZCBpbiB5b3VyIFJEUyBmaWxlCnJlaW1wb3J0X2RmIDwtIDxQVVRfRlVOQ1RJT05fTkFNRT4oZmlsZS5wYXRoKCJyZXN1bHRzIiwgInN0YXRzX2NsZWFuLlJEUyIpKQpgYGAKCkFzIGlzIGdvb2QgcHJhY3RpY2UsIHdlIHdpbGwgZW5kIHRoaXMgc2Vzc2lvbiBieSBwcmludGluZyBvdXQgb3VyIHNlc3Npb24gaW5mby4gCgojIyMgU2Vzc2lvbiBJbmZvCgpgYGB7cn0KIyBQcmludCBvdXQgdGhlIHZlcnNpb25zIGFuZCBwYWNrYWdlcyB3ZSBhcmUgdXNpbmcgaW4gdGhpcyBzZXNzaW9uCnNlc3Npb25JbmZvKCkKYGBgCg==